1   /*
2    * Hibernate, Relational Persistence for Idiomatic Java
3    *
4    * Copyright (c) 2012, Red Hat Inc. or third-party contributors as
5    * indicated by the @author tags or express copyright attribution
6    * statements applied by the authors.  All third-party contributions are
7    * distributed under license by Red Hat Inc.
8    *
9    * This copyrighted material is made available to anyone wishing to use, modify,
10   * copy, or redistribute it subject to the terms and conditions of the GNU
11   * Lesser General Public License, as published by the Free Software Foundation.
12   *
13   * This program is distributed in the hope that it will be useful,
14   * but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
15   * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License
16   * for more details.
17   *
18   * You should have received a copy of the GNU Lesser General Public License
19   * along with this distribution; if not, write to:
20   * Free Software Foundation, Inc.
21   * 51 Franklin Street, Fifth Floor
22   * Boston, MA  02110-1301  USA
23   */
24  package org.hibernate.test.annotations.cascade.multicircle.jpa.sequence;
25  
26  import junit.framework.Assert;
27  import org.junit.After;
28  import org.junit.Before;
29  import org.junit.Test;
30  
31  import org.hibernate.Session;
32  import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase;
33  
34  /**
35   * This test uses a complicated model that requires Hibernate to delay
36   * inserts until non-nullable transient entity dependencies are resolved.
37   *
38   * All IDs are generated from a sequence.
39   *
40   * JPA cascade types are used (javax.persistence.CascadeType)..
41   *
42   * This test uses the following model:
43   *
44   * <code>
45   *     ------------------------------ N G
46   *     |
47   *     |                                1
48   *     |                                |
49   *     |                                |
50   *     |                                N
51   *     |
52   *     |         E N--------------0,1 * F
53   *     |
54   *     |         1                      N
55   *     |         |                      |
56   *     |         |                      |
57   *     1         N                      |
58   *     *                                |
59   *     B * N---1 D * 1------------------
60   *     *
61   *     N         N
62   *     |         |
63   *     |         |
64   *     1         |
65   *               |
66   *     C * 1-----
67   *</code>
68   *
69   * In the diagram, all associations are bidirectional;
70   * assocations marked with '*' cascade persist, save, merge operations to the
71   * associated entities (e.g., B cascades persist to D, but D does not cascade
72   * persist to B);
73   *
74   * b, c, d, e, f, and g are all transient unsaved that are associated with each other.
75   *
76   * When saving b, the entities are added to the ActionQueue in the following order:
77   * c, d (depends on e), f (depends on d, g), e, b, g.
78   *
79   * Entities are inserted in the following order:
80   * c, e, d, b, g, f.
81   */
82  public class MultiCircleJpaCascadeSequenceTest extends BaseCoreFunctionalTestCase {
83  	private B b;
84  	private C c;
85  	private D d;
86  	private E e;
87  	private F f;
88  	private G g;
89  
90  	@Before
91  	public void setup() {
92  		b = new B();
93  		c = new C();
94  		d = new D();
95  		e = new E();
96  		f = new F();
97  		g = new G();
98  
99  		b.getGCollection().add( g );
100 		b.setC( c );
101 		b.setD( d );
102 
103 		c.getBCollection().add( b );
104 		c.getDCollection().add( d );
105 
106 		d.getBCollection().add( b );
107 		d.setC( c );
108 		d.setE( e );
109 		d.getFCollection().add( f );
110 
111 		e.getDCollection().add( d );
112 		e.setF( f );
113 
114 		f.getECollection().add( e );
115 		f.setD( d );
116 		f.setG( g );
117 
118 		g.setB( b );
119 		g.getFCollection().add( f );
120 	}
121 
122 	@After
123 	public void cleanup() {
124 		b.setC( null );
125 		b.setD( null );
126 		b.getGCollection().remove( g );
127 
128 		c.getBCollection().remove( b );
129 		c.getDCollection().remove( d );
130 
131 		d.getBCollection().remove( b );
132 		d.setC( null );
133 		d.setE( null );
134 		d.getFCollection().remove( f );
135 
136 		e.getDCollection().remove( d );
137 		e.setF( null );
138 
139 		f.setD( null );
140 		f.getECollection().remove( e );
141 		f.setG( null );
142 
143 		g.setB( null );
144 		g.getFCollection().remove( f );
145 
146 		Session s = openSession();
147 		s.getTransaction().begin();
148 		b = ( B ) s.merge( b );
149 		c = ( C ) s.merge( c );
150 		d = ( D ) s.merge( d );
151 		e = ( E ) s.merge( e );
152 		f = ( F ) s.merge( f );
153 		g = ( G ) s.merge( g );
154 		s.delete( f );
155 		s.delete( g );
156 		s.delete( b );
157 		s.delete( d );
158 		s.delete( e );
159 		s.delete( c );
160 		s.getTransaction().commit();
161 		s.close();
162 	}
163 
164 	@Test
165 	public void testPersist() {
166 		Session s = openSession();
167 		s.getTransaction().begin();
168 		s.persist( b );
169 		s.getTransaction().commit();
170 		s.close();
171 
172 		check();
173 	}
174 
175 	@Test
176 	public void testMerge() {
177 		Session s = openSession();
178 		s.getTransaction().begin();
179 		b = ( B ) s.merge( b );
180 		c = b.getC();
181 		d = b.getD();
182 		e = d.getE();
183 		f = e.getF();
184 		g = f.getG();
185 		s.getTransaction().commit();
186 		s.close();
187 
188 		check();
189 	}
190 
191 	private void check() {
192 		Session s = openSession();
193 		s.getTransaction().begin();
194 		B bRead = ( B ) s.get( B.class, b.getId() );
195 		Assert.assertEquals( b, bRead );
196 
197 		G gRead = bRead.getGCollection().iterator().next();
198 		Assert.assertEquals( g, gRead );
199 		C cRead = bRead.getC();
200 		Assert.assertEquals( c, cRead );
201 		D dRead = bRead.getD();
202 		Assert.assertEquals( d, dRead );
203 
204 		Assert.assertSame( bRead, cRead.getBCollection().iterator().next() );
205 		Assert.assertSame( dRead, cRead.getDCollection().iterator().next() );
206 
207 		Assert.assertSame( bRead, dRead.getBCollection().iterator().next() );
208 		Assert.assertEquals( cRead, dRead.getC() );
209 		E eRead = dRead.getE();
210 		Assert.assertEquals( e, eRead );
211 		F fRead = dRead.getFCollection().iterator().next();
212 		Assert.assertEquals( f, fRead );
213 
214 		Assert.assertSame( dRead, eRead.getDCollection().iterator().next() );
215 		Assert.assertSame( fRead, eRead.getF() );
216 
217 		Assert.assertSame( eRead, fRead.getECollection().iterator().next() );
218 		Assert.assertSame( dRead, fRead.getD() );
219 		Assert.assertSame( gRead, fRead.getG());
220 
221 		Assert.assertSame( bRead, gRead.getB() );
222 		Assert.assertSame( fRead, gRead.getFCollection().iterator().next() );
223 
224 		s.getTransaction().commit();
225 		s.close();
226 	}
227 
228 	@Override
229 	protected Class[] getAnnotatedClasses() {
230 		return new Class[]{
231 				B.class,
232 				C.class,
233 				D.class,
234 				E.class,
235 				F.class,
236 				G.class
237 		};
238 	}
239 
240 }